import { ApplicationRef, ReflectiveInjector } from '@angular/core';
import { discardPeriodicTasks, fakeAsync, tick } from '@angular/core/testing';
import { SwUpdate } from '@angular/service-worker';
import { Subject } from 'rxjs';

import { Logger } from 'app/shared/logger.service';
import { SwUpdatesService } from './sw-updates.service';


describe('SwUpdatesService', () => {
  let injector: ReflectiveInjector;
  let appRef: MockApplicationRef;
  let service: SwUpdatesService;
  let swu: MockSwUpdate;
  let checkInterval: number;

  // Helpers
  // NOTE:
  //   Because `SwUpdatesService` uses the `interval` operator, it needs to be instantiated and
  //   destroyed inside the `fakeAsync` zone (when `fakeAsync` is used for the test). Thus, we can't
  //   run `setup()`/`tearDown()` in `beforeEach()`/`afterEach()` blocks. We use the `run()` helper
  //   to call them inside each test's zone.
  const setup = (isSwUpdateEnabled: boolean) => {
    injector = ReflectiveInjector.resolveAndCreate([
      { provide: ApplicationRef, useClass: MockApplicationRef },
      { provide: Logger, useClass: MockLogger },
      { provide: SwUpdate, useFactory: () => new MockSwUpdate(isSwUpdateEnabled) },
      SwUpdatesService
    ]);

    appRef = injector.get(ApplicationRef);
    service = injector.get(SwUpdatesService);
    swu = injector.get(SwUpdate);
    checkInterval = (service as any).checkInterval;
  };
  const finalize = () => service.ngOnDestroy();
  const run = (specFn: VoidFunction, isSwUpdateEnabled = true) => () => {
    setup(isSwUpdateEnabled);
    specFn();
    finalize();
  };


  it('should create', run(() => {
    expect(service).toBeTruthy();
  }));

  it('should start checking for updates when instantiated (once the app stabilizes)', run(() => {
    expect(swu.checkForUpdate).not.toHaveBeenCalled();

    appRef.isStable.next(false);
    expect(swu.checkForUpdate).not.toHaveBeenCalled();

    appRef.isStable.next(true);
    expect(swu.checkForUpdate).toHaveBeenCalled();
  }));

  it('should periodically check for updates', fakeAsync(run(() => {
    appRef.isStable.next(true);
    swu.checkForUpdate.calls.reset();

    tick(checkInterval);
    expect(swu.checkForUpdate).toHaveBeenCalledTimes(1);

    tick(checkInterval);
    expect(swu.checkForUpdate).toHaveBeenCalledTimes(2);

    appRef.isStable.next(false);

    tick(checkInterval);
    expect(swu.checkForUpdate).toHaveBeenCalledTimes(3);

    discardPeriodicTasks();
  })));

  it('should activate available updates immediately', fakeAsync(run(() => {
    appRef.isStable.next(true);
    expect(swu.activateUpdate).not.toHaveBeenCalled();

    swu.$$availableSubj.next({available: {hash: 'foo'}});
    expect(swu.activateUpdate).toHaveBeenCalled();
  })));

  it('should keep periodically checking for updates even after one is available/activated', fakeAsync(run(() => {
    appRef.isStable.next(true);
    swu.checkForUpdate.calls.reset();

    tick(checkInterval);
    expect(swu.checkForUpdate).toHaveBeenCalledTimes(1);

    swu.$$availableSubj.next({available: {hash: 'foo'}});

    tick(checkInterval);
    expect(swu.checkForUpdate).toHaveBeenCalledTimes(2);

    tick(checkInterval);
    expect(swu.checkForUpdate).toHaveBeenCalledTimes(3);

    discardPeriodicTasks();
  })));

  it('should emit on `updateActivated` when an update has been activated', run(() => {
    const activatedVersions: (string|undefined)[] = [];
    service.updateActivated.subscribe(v => activatedVersions.push(v));

    swu.$$availableSubj.next({available: {hash: 'foo'}});
    swu.$$activatedSubj.next({current: {hash: 'bar'}});
    swu.$$availableSubj.next({available: {hash: 'baz'}});
    swu.$$activatedSubj.next({current: {hash: 'qux'}});

    expect(activatedVersions).toEqual(['bar', 'qux']);
  }));

  describe('when `SwUpdate` is not enabled', () => {
    const runDeactivated = (specFn: VoidFunction) => run(specFn, false);

    it('should not check for updates', fakeAsync(runDeactivated(() => {
      appRef.isStable.next(true);

      tick(checkInterval);
      tick(checkInterval);

      swu.$$availableSubj.next({available: {hash: 'foo'}});
      swu.$$activatedSubj.next({current: {hash: 'bar'}});

      tick(checkInterval);
      tick(checkInterval);

      expect(swu.checkForUpdate).not.toHaveBeenCalled();
    })));

    it('should not activate available updates', fakeAsync(runDeactivated(() => {
      swu.$$availableSubj.next({available: {hash: 'foo'}});
      expect(swu.activateUpdate).not.toHaveBeenCalled();
    })));

    it('should never emit on `updateActivated`', runDeactivated(() => {
      const activatedVersions: (string|undefined)[] = [];
      service.updateActivated.subscribe(v => activatedVersions.push(v));

      swu.$$availableSubj.next({available: {hash: 'foo'}});
      swu.$$activatedSubj.next({current: {hash: 'bar'}});
      swu.$$availableSubj.next({available: {hash: 'baz'}});
      swu.$$activatedSubj.next({current: {hash: 'qux'}});

      expect(activatedVersions).toEqual([]);
    }));
  });

  describe('when destroyed', () => {
    it('should not schedule a new check for update (after current check)', fakeAsync(run(() => {
      appRef.isStable.next(true);
      expect(swu.checkForUpdate).toHaveBeenCalled();

      service.ngOnDestroy();
      swu.checkForUpdate.calls.reset();

      tick(checkInterval);
      tick(checkInterval);

      expect(swu.checkForUpdate).not.toHaveBeenCalled();
    })));

    it('should not schedule a new check for update (after activating an update)', fakeAsync(run(() => {
      appRef.isStable.next(true);
      expect(swu.checkForUpdate).toHaveBeenCalled();

      service.ngOnDestroy();
      swu.checkForUpdate.calls.reset();

      swu.$$availableSubj.next({available: {hash: 'foo'}});
      swu.$$activatedSubj.next({current: {hash: 'baz'}});

      tick(checkInterval);
      tick(checkInterval);

      expect(swu.checkForUpdate).not.toHaveBeenCalled();
    })));

    it('should not activate available updates', fakeAsync(run(() => {
      service.ngOnDestroy();
      swu.$$availableSubj.next({available: {hash: 'foo'}});

      expect(swu.activateUpdate).not.toHaveBeenCalled();
    })));

    it('should stop emitting on `updateActivated`', run(() => {
      const activatedVersions: (string|undefined)[] = [];
      service.updateActivated.subscribe(v => activatedVersions.push(v));

      swu.$$availableSubj.next({available: {hash: 'foo'}});
      swu.$$activatedSubj.next({current: {hash: 'bar'}});
      service.ngOnDestroy();
      swu.$$availableSubj.next({available: {hash: 'baz'}});
      swu.$$activatedSubj.next({current: {hash: 'qux'}});

      expect(activatedVersions).toEqual(['bar']);
    }));
  });
});

// Mocks
class MockApplicationRef {
  isStable = new Subject<boolean>();
}

class MockLogger {
  log = jasmine.createSpy('MockLogger.log');
}

class MockSwUpdate {
  $$availableSubj = new Subject<{available: {hash: string}}>();
  $$activatedSubj = new Subject<{current: {hash: string}}>();

  available = this.$$availableSubj.asObservable();
  activated = this.$$activatedSubj.asObservable();

  activateUpdate = jasmine.createSpy('MockSwUpdate.activateUpdate')
                          .and.callFake(() => Promise.resolve());

  checkForUpdate = jasmine.createSpy('MockSwUpdate.checkForUpdate')
                          .and.callFake(() => Promise.resolve());

  constructor(public isEnabled: boolean) {}
}
